/* -*- Mode: C++; tab-width: 2; indent-tabs-mode: nil; c-basic-offset: 2 -*- */
/* ***** BEGIN LICENSE BLOCK *****
 * Version: MPL 1.1/GPL 2.0/LGPL 2.1
 *
 * The contents of this file are subject to the Mozilla Public License Version
 * 1.1 (the "License"); you may not use this file except in compliance with
 * the License. You may obtain a copy of the License at
 * http://www.mozilla.org/MPL/
 *
 * Software distributed under the License is distributed on an "AS IS" basis,
 * WITHOUT WARRANTY OF ANY KIND, either express or implied. See the License
 * for the specific language governing rights and limitations under the
 * License.
 *
 * The Original Code is JavaScript Engine testing utilities.
 *
 * The Initial Developer of the Original Code is
 * Netscape Communication Corporation.
 * Portions created by the Initial Developer are Copyright (C) 1998
 * the Initial Developer. All Rights Reserved.
 *
 * Contributor(s):
 *
 * Alternatively, the contents of this file may be used under the terms of
 * either the GNU General Public License Version 2 or later (the "GPL"), or
 * the GNU Lesser General Public License Version 2.1 or later (the "LGPL"),
 * in which case the provisions of the GPL or the LGPL are applicable instead
 * of those above. If you wish to allow use of your version of this file only
 * under the terms of either the GPL or the LGPL, and not to allow others to
 * use your version of this file under the terms of the MPL, indicate your
 * decision by deleting the provisions above and replace them with the notice
 * and other provisions required by the GPL or the LGPL. If you do not delete
 * the provisions above, a recipient may use your version of this file under
 * the terms of any one of the MPL, the GPL or the LGPL.
 *
 * ***** END LICENSE BLOCK ***** */

gTestfile = 'split-002.js';

/**
 *  File Name:          String/split-002.js
 *  ECMA Section:       15.6.4.9
 *  Description:        Based on ECMA 2 Draft 7 February 1999
 *
 *  Author:             christine@netscape.com
 *  Date:               19 February 1999
 */

/*
 * Since regular expressions have been part of JavaScript since 1.2, there
 * are already tests for regular expressions in the js1_2/regexp folder.
 *
 * These new tests try to supplement the existing tests, and verify that
 * our implementation of RegExp conforms to the ECMA specification, but
 * does not try to be as exhaustive as in previous tests.
 *
 * The [,limit] argument to String.split is new, and not covered in any
 * existing tests.
 *
 * String.split cases are covered in ecma/String/15.5.4.8-*.js.
 * String.split where separator is a RegExp are in
 * js1_2/regexp/string_split.js
 *
 */

var SECTION = "ecma_2/String/split-002.js";
var VERSION = "ECMA_2";
var TITLE   = "String.prototype.split( regexp, [,limit] )";

startTest();

// the separator is not supplied
// separator is undefined
// separator is an empty string

//    AddSplitCases( "splitme", "", "''", ["s", "p", "l", "i", "t", "m", "e"] );
//    AddSplitCases( "splitme", new RegExp(), "new RegExp()", ["s", "p", "l", "i", "t", "m", "e"] );

// separator is an empty regexp
// separator is not supplied

CompareSplit( "hello", "ll" );

CompareSplit( "hello", "l" );
CompareSplit( "hello", "x" );
CompareSplit( "hello", "h" );
CompareSplit( "hello", "o" );
CompareSplit( "hello", "hello" );
CompareSplit( "hello", undefined );

CompareSplit( "hello", "");
CompareSplit( "hello", "hellothere" );

CompareSplit( new String("hello" ) );


Number.prototype.split = String.prototype.split;

CompareSplit( new Number(100111122133144155), 1 );
CompareSplitWithLimit(new Number(100111122133144155), 1, 1 );

CompareSplitWithLimit(new Number(100111122133144155), 1, 2 );
CompareSplitWithLimit(new Number(100111122133144155), 1, 0 );
CompareSplitWithLimit(new Number(100111122133144155), 1, 100 );
CompareSplitWithLimit(new Number(100111122133144155), 1, void 0 );
CompareSplitWithLimit(new Number(100111122133144155), 1, Math.pow(2,32)-1 );
CompareSplitWithLimit(new Number(100111122133144155), 1, "boo" );
CompareSplitWithLimit(new Number(100111122133144155), 1, -(Math.pow(2,32)-1) );
CompareSplitWithLimit( "hello", "l", NaN );
CompareSplitWithLimit( "hello", "l", 0 );
CompareSplitWithLimit( "hello", "l", 1 );
CompareSplitWithLimit( "hello", "l", 2 );
CompareSplitWithLimit( "hello", "l", 3 );
CompareSplitWithLimit( "hello", "l", 4 );


/*
  CompareSplitWithLimit( "hello", "ll", 0 );
  CompareSplitWithLimit( "hello", "ll", 1 );
  CompareSplitWithLimit( "hello", "ll", 2 );
  CompareSplit( "", " " );
  CompareSplit( "" );
*/

// separartor is a regexp
// separator regexp value global setting is set
// string is an empty string
// if separator is an empty string, split each by character

// this is not a String object

// limit is not a number
// limit is undefined
// limit is larger than 2^32-1
// limit is a negative number

test();

function CompareSplit( string, separator ) {
  split_1 = string.split( separator );
  split_2 = string_split( string, separator );

  AddTestCase(
    "( " + string +".split(" + separator + ") ).length" ,
    split_2.length,
    split_1.length );

  var limit = split_1.length > split_2.length ?
    split_1.length : split_2.length;

  for ( var split_item = 0; split_item < limit; split_item++ ) {
    AddTestCase(
      string + ".split(" + separator + ")["+split_item+"]",
      split_2[split_item],
      split_1[split_item] );
  }
}

function CompareSplitWithLimit( string, separator, splitlimit ) {
  split_1 = string.split( separator, splitlimit );
  split_2 = string_split( string, separator, splitlimit );

  AddTestCase(
    "( " + string +".split(" + separator + ", " + splitlimit+") ).length" ,
    split_2.length,
    split_1.length );

  var limit = split_1.length > split_2.length ?
    split_1.length : split_2.length;

  for ( var split_item = 0; split_item < limit; split_item++ ) {
    AddTestCase(
      string + ".split(" + separator  + ", " + splitlimit+")["+split_item+"]",
      split_2[split_item],
      split_1[split_item] );
  }
}

function string_split ( __this, separator, limit ) {
  var S = String(__this );					  // 1

  var A = new Array();                          // 2

  if ( limit == undefined ) {                   // 3
    lim = Math.pow(2, 31 ) -1;
  } else {
    lim = ToUint32( limit );
  }

  var s = S.length;                              // 4
  var p = 0;                                     // 5

  if  ( separator == undefined ) {              // 8
    A[0] = S;
    return A;
  }

  if ( separator.constructor == RegExp )         // 6
    R = separator;
  else
    R = separator.toString();

  if (lim == 0) return A;                       // 7

  if  ( separator == undefined ) {              // 8
    A[0] = S;
    return A;
  }

  if (s == 0) {		                          // 9
    z = SplitMatch(R, S, 0);
    if (z != false) return A;
    A[0] = S;
    return A;
  }

  var q = p;									  // 10
loop:
  while (true ) {
	
    if ( q == s ) break;					  // 11

    z = SplitMatch(R, S, q);                  // 12

//print("Returned ", z);

    if (z != false) {							// 13
      e = z.endIndex;							// 14
      cap = z.captures;						// 14
      if (e != p) {							// 15
//print("S = ", S, ", p = ", p, ", q = ", q);
	T = S.slice(p, q);					// 16
//print("T = ", T);
	A[A.length] = T;					// 17
	if (A.length == lim) return A;		// 18
	p = e;								// 19
	i = 0;								// 20
	while (true) {						// 25
	  if (i == cap.length) {              // 21
	    q = p;                          // 10
	    continue loop;
	  }
	  i = i + 1;							// 22
	  A[A.length] = cap[i]				// 23
	    if (A.length == lim) return A;		// 24
	}
      }
    }

    q = q + 1;                               // 26
  }

  T = S.slice(p, q);
  A[A.length] = T;
  return A;
}

function SplitMatch(R, S, q)
{
  if (R.constructor == RegExp) {			// 1
    var reResult = R.match(S, q);		// 8
    if (reResult == undefined)
      return false;
    else {
      a = new Array(reResult.length - 1);
      for (var i = 1; i < reResult.length; i++)
	a[a.length] = reResult[i];
      return { endIndex : reResult.index + reResult[0].length, captures : cap };
    }
  }
  else {
    var r = R.length;					// 2
    s = S.length;						// 3
    if ((q + r) > s) return false;		// 4
    for (var i = 0; i < r; i++) {
//print("S.charAt(", q + i, ") = ", S.charAt(q + i), ", R.charAt(", i, ") = ", R.charAt(i));
      if (S.charAt(q + i) != R.charAt(i))			// 5
	return false;
    }
    cap = new Array();								// 6
    return { endIndex : q + r, captures : cap };	// 7
  }
}

function ToUint32( n ) {
  n = Number( n );
  var sign = ( n < 0 ) ? -1 : 1;

  if ( Math.abs( n ) == 0
       || Math.abs( n ) == Number.POSITIVE_INFINITY
       || n != n) {
    return 0;
  }
  n = sign * Math.floor( Math.abs(n) )

    n = n % Math.pow(2,32);

  if ( n < 0 ){
    n += Math.pow(2,32);
  }

  return ( n );
}
